home *** CD-ROM | disk | FTP | other *** search
/ The CICA Windows Explosion! / The CICA Windows Explosion! - Disc 2.iso / programr / fileutil.zip / CMP.C < prev    next >
C/C++ Source or Header  |  1992-02-22  |  14KB  |  590 lines

  1. /* cmp -- compare two files.
  2.    Copyright (C) 1988, 1990 Free Software Foundation, Inc.
  3.  
  4.    This program is free software; you can redistribute it and/or modify
  5.    it under the terms of the GNU General Public License as published by
  6.    the Free Software Foundation; either version 1, or (at your option)
  7.    any later version.
  8.  
  9.    This program is distributed in the hope that it will be useful,
  10.    but WITHOUT ANY WARRANTY; without even the implied warranty of
  11.    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  12.    GNU General Public License for more details.
  13.  
  14.    You should have received a copy of the GNU General Public License
  15.    along with this program; if not, write to the Free Software
  16.    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.  */
  17.  
  18. /* MS-DOS port (c) 1990 by Thorsten Ohl, ohl@gnu.ai.mit.edu
  19.    This port is also distributed under the terms of the
  20.    GNU General Public License as published by the
  21.    Free Software Foundation.
  22.  
  23.    Please note that this file is not identical to the
  24.    original GNU release, you should have received this
  25.    code as patch to the official release.  */
  26.  
  27. #ifdef MSDOS
  28. static char RCS_Id[] =
  29.   "$Header: e:/gnu/fileutil/RCS/cmp.c 1.4.0.2 90/09/19 11:17:46 tho Exp $";
  30.  
  31. static char Program_Id[] = "cmp";
  32. static char RCS_Revision[] = "$Revision: 1.4.0.2 $";
  33.  
  34. #define VERSION \
  35.   "GNU %s, Version %.*s (compiled %s %s for MS-DOS)\n", Program_Id, \
  36.   (sizeof RCS_Revision - 14), (RCS_Revision + 11), __DATE__, __TIME__
  37.  
  38. #define COPYING \
  39.   "This is free software, distributed under the terms of the\n" \
  40.   "GNU General Public License.  For details, see the file COPYING.\n"
  41. #endif /* MSDOS */
  42.  
  43. /* Differences from the Unix cmp:
  44.    * 6 - 40 - oo times faster.
  45.    * The file name `-' is always the standard input. If one file name
  46.      is omitted, the standard input is used as well.
  47.    * -c option to print the differing characters like `cat -t'
  48.      (except that newlines are printed as `^J'), with or without -l.
  49.  
  50.    By Torbjorn Granlund and David MacKenzie. */
  51.  
  52. #include <stdio.h>
  53. #include <getopt.h>
  54. #include <errno.h>
  55. #include <sys/types.h>
  56. #include "system.h"
  57.  
  58. #ifdef STDC_HEADERS
  59. #include <stdlib.h>
  60. #else
  61. char *malloc ();
  62.  
  63. extern int errno;
  64. #endif
  65.  
  66. #ifdef MSDOS
  67. #include <io.h>
  68. #include <gnulib.h>
  69.  
  70. extern void main (int, char **);
  71. extern void usage (char *);
  72. extern void cmp (void);
  73. extern int bcmp2 (char *, char *);
  74. extern int bcmp_cnt (int *count, char *p1, char *p2, unsigned char c);
  75. extern int bread (int, char *, int);
  76. extern void printc (FILE *fs, int width, unsigned int c);
  77.  
  78. #else /* not MSDOS */
  79.  
  80. #define max(h, i)    ((h) > (i) ? (h) : (i))
  81. #define min(l, o)    ((l) < (o) ? (l) : (o))
  82.  
  83. int bcmp_cnt ();
  84. int bcmp2 ();
  85. void cmp ();
  86. int bread ();
  87. void printc ();
  88. void error ();
  89.  
  90. #endif /* not MSDOS */
  91.  
  92. /* Name under which this program was invoked.  */
  93.  
  94. char *program_name;
  95.  
  96. /* Filenames of the compared files.  */
  97.  
  98. char *file1;
  99. char *file2;
  100.  
  101. /* File descriptors of the files.  */
  102.  
  103. int file1_desc;
  104. int file2_desc;
  105.  
  106. /* Read buffers for the files.  */
  107.  
  108. char *buf1;
  109. char *buf2;
  110.  
  111. /* Optimal block size for the files.  */
  112.  
  113. int buf_size;
  114.  
  115. /* Output format:
  116.    0 to print the offset and line number of the first differing bytes
  117.    'l' to print the (decimal) offsets and (octal) values of all differing bytes
  118.    's' to only return an exit status indicating whether the files differ */
  119.  
  120. int flag = 0;
  121.  
  122. /* If nonzero, print values of bytes quoted like cat -t does. */
  123. int flag_print_chars = 0;
  124.  
  125. struct option long_options[] =
  126. {
  127. #ifdef MSDOS
  128.   {"copying", 0, NULL, 30},
  129.   {"version", 0, NULL, 31},
  130. #endif
  131.   {"show-chars", 0, &flag_print_chars, 1},
  132.   {"silent", 0, &flag, 's'},
  133.   {"quiet", 0, &flag, 's'},
  134.   {"verbose", 0, &flag, 'l'},
  135.   {NULL, 0, NULL, 0}
  136. };
  137.  
  138. void
  139. usage (reason)
  140.      char *reason;
  141. {
  142.   if (reason != NULL)
  143.     fprintf (stderr, "%s: %s\n", program_name, reason);
  144.  
  145. #ifdef MSDOS
  146.   fprintf (stderr, "\
  147. Usage: %s [-cls] [+show-chars] [+verbose] [+silent] [+quiet]\n\
  148.            [+copying] [+version] file1 [file2]\n",
  149.        program_name);
  150. #else /* not MSDOS */
  151.   fprintf (stderr, "\
  152. Usage: %s [-cls] [+show-chars] [+verbose] [+silent] [+quiet] file1 [file2]\n",
  153.        program_name);
  154. #endif /* not MSDOS */
  155.  
  156.   exit (2);
  157. }
  158.  
  159. void
  160. main (argc, argv)
  161.      int argc;
  162.      char *argv[];
  163. {
  164.   int c;
  165.   struct stat stat_buf1, stat_buf2;
  166.   int ind;
  167.  
  168.   program_name = argv[0];
  169.  
  170.   /* If an argument is omitted, default to the standard input.  */
  171.  
  172.   file1 = "-";
  173.   file2 = "-";
  174.   file1_desc = fileno (stdin);
  175.   file2_desc = fileno (stdin);
  176.  
  177.   /* Parse command line options.  */
  178.  
  179.   while ((c = getopt_long (argc, argv, "cls", long_options, &ind)) != EOF)
  180.     switch (c)
  181.       {
  182.       case 0:
  183.     break;
  184.  
  185.       case 'c':
  186.     flag_print_chars = 1;
  187.     break;
  188.  
  189.       case 'l':
  190.       case 's':
  191.     flag = c;
  192.     break;
  193.  
  194. #ifdef MSDOS
  195.       case 30:
  196.     fprintf (stderr, COPYING);
  197.     exit (0);
  198.     break;
  199.  
  200.       case 31:
  201.     fprintf (stderr, VERSION);
  202.     exit (0);
  203.     break;
  204. #endif
  205.  
  206.       default:
  207.     usage ((char *) 0);
  208.       }
  209.  
  210.   if (optind < argc)
  211.     file1 = argv[optind++];
  212.  
  213.   if (optind < argc)
  214.     file2 = argv[optind++];
  215.  
  216.   if (optind < argc)
  217.     usage ("extra arguments");
  218.  
  219.   if (strcmp (file1, "-"))
  220.     {
  221.       file1_desc = open (file1, O_RDONLY);
  222.       if (file1_desc < 0)
  223.     {
  224.       if (flag == 's')
  225.         exit (2);
  226.       else
  227.         error (2, errno, "%s", file1);
  228.     }
  229.     }
  230.  
  231.   if (strcmp (file2, "-"))
  232.     {
  233.       file2_desc = open (file2, O_RDONLY);
  234.       if (file2_desc < 0)
  235.     {
  236.       if (flag == 's')
  237.         exit (2);
  238.       else
  239.         error (2, errno, "%s", file2);
  240.     }
  241.     }
  242.  
  243.   if (file1_desc == file2_desc)
  244.     usage ("at least one filename should be specified");
  245.  
  246.   if (fstat (file1_desc, &stat_buf1) < 0)
  247.     error (2, errno, "%s", file1);
  248.   if (fstat (file2_desc, &stat_buf2) < 0)
  249.     error (2, errno, "%s", file2);
  250.  
  251.   /* If both the input descriptors are associated with plain files,
  252.      we can make the job simpler in some cases.  */
  253.  
  254.   if ((stat_buf1.st_mode & S_IFMT) == S_IFREG
  255.       && (stat_buf2.st_mode & S_IFMT) == S_IFREG)
  256.     {
  257.       /* Find out if the files are links to the same inode, and therefore
  258.      identical.  */
  259.  
  260.       if (stat_buf1.st_dev == stat_buf2.st_dev
  261.       && stat_buf1.st_ino == stat_buf2.st_ino)
  262.     exit (0);
  263.  
  264.       /* If output is redirected to "/dev/null", we may assume `-s'.  */
  265.  
  266.       if (flag != 's')
  267.     {
  268.       struct stat sb;
  269.       dev_t nulldev;
  270.       ino_t nullino;
  271.  
  272.       if (stat ("/dev/null", &sb) == 0)
  273.         {
  274.           nulldev = sb.st_dev;
  275.           nullino = sb.st_ino;
  276.           if (fstat (1, &sb) == 0
  277.           && sb.st_dev == nulldev && sb.st_ino == nullino)
  278.         flag = 's';
  279.         }
  280.     }
  281.  
  282.       /* If only a return code is needed, conclude that
  283.      the files differ if they have different sizes.  */
  284.  
  285.       if (flag == 's' && stat_buf1.st_size != stat_buf2.st_size)
  286.     exit (1);
  287.     }
  288.  
  289.   /* Get the optimal block size of the files.  */
  290.  
  291.   buf_size = max (ST_BLKSIZE (stat_buf1), ST_BLKSIZE (stat_buf2));
  292.  
  293.   /* Allocate buffers, with space for sentinels at the end.  */
  294.  
  295.   buf1 = malloc (buf_size + sizeof (long));
  296.   buf2 = malloc (buf_size + sizeof (long));
  297.   if (buf1 == NULL || buf2 == NULL)
  298.     error (2, 0, "virtual memory exhausted");
  299.  
  300. #ifdef MSDOS
  301.   setmode (file1_desc, O_BINARY);
  302.   setmode (file2_desc, O_BINARY);
  303. #endif
  304.  
  305.   cmp ();
  306. }
  307.  
  308. /* Compare the two files already open on `file1_desc' and `file2_desc',
  309.    using `buf1' and `buf2'.  */
  310.  
  311. void
  312. cmp ()
  313. {
  314.   long line_number = 1;        /* Line number (1...) of first difference. */
  315.   long char_number = 1;        /* Offset (1...) in files of 1st difference. */
  316.   int read1, read2;        /* Number of bytes read from each file. */
  317.   int first_diff;        /* Offset (0...) in buffers of 1st diff. */
  318.   int smaller;            /* The lesser of `read1' and `read2'. */
  319.   int exit_status = 0;
  320.  
  321.   do
  322.     {
  323.       read1 = bread (file1_desc, buf1, buf_size);
  324.       if (read1 < 0)
  325.     error (2, errno, "%s", file1);
  326.       read2 = bread (file2_desc, buf2, buf_size);
  327.       if (read2 < 0)
  328.     error (2, errno, "%s", file2);
  329.  
  330.       /* Insert sentinels for the block compare.  */
  331.  
  332.       buf1[read1] = ~buf2[read1];
  333.       buf2[read2] = ~buf1[read2];
  334.  
  335.       if (flag == 0)
  336.     {
  337.       int cnt;
  338.  
  339.       /* If the line number should be written for differing files,
  340.          compare the blocks and count the number of newlines
  341.          simultaneously.  */
  342.       first_diff = bcmp_cnt (&cnt, buf1, buf2, '\n');
  343.       line_number += cnt;
  344.     }
  345.       else
  346.     {
  347.       first_diff = bcmp2 (buf1, buf2);
  348.     }
  349.  
  350.       if (flag != 'l')
  351.     char_number += first_diff;
  352.       else
  353.     smaller = min (read1, read2);
  354.  
  355.       if (first_diff < read1 && first_diff < read2)
  356.     {
  357.       switch (flag)
  358.         {
  359.         case 0:
  360.           /* This format is a proposed POSIX standard.  */
  361.           printf ("%s %s differ: char %ld, line %ld",
  362.               file1, file2, char_number, line_number);
  363.           if (flag_print_chars)
  364.         {
  365.           printf (" is");
  366.           printf (" %3o ",
  367.               (unsigned) (unsigned char) buf1[first_diff]);
  368.           printc (stdout, 0,
  369.               (unsigned) (unsigned char) buf1[first_diff]);
  370.           printf (" %3o ",
  371.               (unsigned) (unsigned char) buf2[first_diff]);
  372.           printc (stdout, 0,
  373.               (unsigned) (unsigned char) buf2[first_diff]);
  374.         }
  375.           putchar ('\n');
  376.           /* Fall through. */
  377.         case 's':
  378.           exit (1);
  379.         case 'l':
  380.           while (first_diff < smaller)
  381.         {
  382.           if (buf1[first_diff] != buf2[first_diff])
  383.             {
  384.               if (flag_print_chars)
  385.             {
  386.               printf ("%6ld", first_diff + char_number);
  387.               printf (" %3o ",
  388.                   (unsigned) (unsigned char) buf1[first_diff]);
  389.               printc (stdout, 4,
  390.                   (unsigned) (unsigned char) buf1[first_diff]);
  391.               printf (" %3o ",
  392.                   (unsigned) (unsigned char) buf2[first_diff]);
  393.               printc (stdout, 0,
  394.                   (unsigned) (unsigned char) buf2[first_diff]);
  395.               putchar ('\n');
  396.             }
  397.               else
  398.             /* This format is a proposed POSIX standard. */
  399.             printf ("%6ld %3o %3o\n",
  400.                 first_diff + char_number,
  401.                 (unsigned) (unsigned char) buf1[first_diff],
  402.                 (unsigned) (unsigned char) buf2[first_diff]);
  403.             }
  404.           first_diff++;
  405.         }
  406.           exit_status = 1;
  407.           break;
  408.         }
  409.     }
  410.  
  411.       if (flag == 'l')
  412.     char_number += smaller;
  413.  
  414.       if (read1 != read2)
  415.     {
  416.       switch (flag)
  417.         {
  418.         case 0:
  419.         case 'l':
  420.           /* This format is a proposed POSIX standard. */
  421.           printf ("%s: EOF on %s\n",
  422.               program_name, read1 < read2 ? file1 : file2);
  423.           break;
  424.         case 's':
  425.           break;
  426.         }
  427.       exit (1);
  428.     }
  429.     }
  430.   while (read1);
  431.   exit (exit_status);
  432. }
  433.  
  434. /* Compare two blocks of memory P1 and P2 until they differ,
  435.    and count the number of occurences of the character C in the common
  436.    part of P1 and P2.
  437.    Assumes that P1 and P2 are aligned at long addresses!
  438.    If the blocks are not guaranteed to be different, put sentinels at the ends
  439.    of the blocks before calling this function.
  440.    Return the offset of the first byte that differs.
  441.    Place the count at the address pointed to by COUNT.  */
  442.  
  443. int
  444. bcmp_cnt (count, p1, p2, c)
  445.      int *count;
  446.      char *p1, *p2;
  447.      unsigned char c;
  448. {
  449.   long w;            /* Word for counting C. */
  450.   long i1, i2;            /* One word from each buffer to compare. */
  451.   long *p1i, *p2i;        /* Pointers into each buffer. */
  452.   char *p1c, *p2c;        /* Pointers for finding exact address. */
  453.   unsigned int cnt = 0;        /* Number of occurrences of C. */
  454.   long cccc;            /* C, four times. */
  455.   long m0, m1, m2, m3;        /* Bitmasks for counting C. */
  456.  
  457.   cccc = ((long) c << 24) | ((long) c << 16) | ((long) c << 8) | ((long) c);
  458.  
  459.   m0 = 0xff;
  460.   m1 = 0xff00;
  461.   m2 = 0xff0000;
  462.   m3 = 0xff000000;
  463.  
  464.   p1i = (long *) p1;
  465.   p2i = (long *) p2;
  466.  
  467.   /* Find the rough position of the first difference by reading long ints,
  468.      not bytes.  */
  469.  
  470.   i1 = *p1i++;
  471.   i2 = *p2i++;
  472.   while (i1 == i2)
  473.     {
  474.       w = i1 ^ cccc;
  475.       cnt += (w & m0) == 0;
  476.       cnt += (w & m1) == 0;
  477.       cnt += (w & m2) == 0;
  478.       cnt += (w & m3) == 0;
  479.       i1 = *p1i++;
  480.       i2 = *p2i++;
  481.     }
  482.  
  483.   /* Find out the exact differing position (endianess independant).  */
  484.  
  485.   p1c = (char *) (p1i - 1);
  486.   p2c = (char *) (p2i - 1);
  487.   while (*p1c == *p2c)
  488.     {
  489.       cnt += c == *p1c;
  490.       p1c++;
  491.       p2c++;
  492.     }
  493.  
  494.   *count = cnt;
  495.   return p1c - p1;
  496. }
  497.  
  498.  
  499. /* Compare two blocks of memory P1 and P2 until they differ.
  500.    Assumes that P1 and P2 are aligned at long addresses!
  501.    If the blocks are not guaranteed to be different, put sentinels at the ends
  502.    of the blocks before calling this function.
  503.    Return the offset of the first byte that differs.  */
  504.  
  505. int
  506. bcmp2 (p1, p2)
  507.      char *p1, *p2;
  508. {
  509.   long *i1, *i2;
  510.   char *c1, *c2;
  511.  
  512.   /* Find the rough position of the first difference by reading long ints,
  513.      not bytes.  */
  514.  
  515.   for (i1 = (long *) p1, i2 = (long *) p2; *i1++ == *i2++;)
  516.     ;
  517.  
  518.   /* Find out the exact differing position (endianess independant).  */
  519.  
  520.   for (c1 = (char *) (i1 - 1), c2 = (char *) (i2 - 1); *c1 == *c2; c1++, c2++)
  521.     ;
  522.  
  523.   return c1 - p1;
  524. }
  525.  
  526. /* Read NCHARS bytes from descriptor FD into BUF.
  527.    Return the number of characters successfully read.  */
  528.  
  529. int
  530. bread (fd, buf, nchars)
  531.      int fd;
  532.      char *buf;
  533.      int nchars;
  534. {
  535.   char *bp = buf;
  536.   int nread;
  537.  
  538.   for (;;)
  539.     {
  540.       nread = read (fd, bp, nchars);
  541.       if (nread < 0)
  542.     return -1;
  543.       bp += nread;
  544.       if (nread == nchars || nread == 0)
  545.     break;
  546.       nchars -= nread;
  547.     }
  548.   return bp - buf;
  549. }
  550.  
  551. /* Print character C on stream FS, making nonvisible characters
  552.    visible by quoting like cat -t does.
  553.    Pad with spaces on the right to WIDTH characters.  */
  554.  
  555. void
  556. printc (fs, width, c)
  557.      FILE *fs;
  558.      int width;
  559.      unsigned c;
  560. {
  561.   if (c >= 128)
  562.     {
  563.       putc ('M', fs);
  564.       putc ('-', fs);
  565.       c -= 128;
  566.       width -= 2;
  567.     }
  568.   if (c < 32)
  569.     {
  570.       putc ('^', fs);
  571.       c += 64;
  572.       --width;
  573.     }
  574.   else if (c == 127)
  575.     {
  576.       putc ('^', fs);
  577.       c = '?';
  578.       --width;
  579.     }
  580.  
  581. #ifdef MSDOS
  582.   putc ((int) c, fs);
  583. #else
  584.   putc (c, fs);
  585. #endif
  586.  
  587.   while (--width > 0)
  588.     putc (' ', fs);
  589. }
  590.